Java 中的 String 类

作者:xuguoqing07。希望能把所学到的内容记录下来,同时能帮助到他人。文中若有错误以及各位若有新的想法,还请指点一二,在下感激不尽。

String 类:

(1)String str=”abc”;

如果方法区中的常量池里存在 abc,则 str 里面放的是此 abc 的地址,str直接指向方法区中的常量池;如果方法区中的常量池里没有 abc,那么会在常量池中新建一个 abc,然后再把这个 abc 的地址给 str。总之,str 直接指向方法区中的常量池。

(2)String str1=new String(“abc”);

str1 (里面存的是 new 出来的对象的地址)指向堆内存中的对象。
在这里需要说明一下,String 底层是 char 类型的数组,同时 String 是不可变的字符序列。根据下面的内存图(画艺不精……)来解释一下(1)、(2)两个语句。

1

由上图可以看出 String str="abc"中的 str1 直接指向方法区中的常量池里的 abc ,而图中 String str3=new String("abc") 中的 str3 里存放的新 new 出来的对象的地址,即 str3 直接指向新建出来的对象(为了方便,暂且叫做对象 1)。但是 abc 却不存放在对象 1 里,对象 1 里实际上存在一个数组(叫做 value[ ] 数组),该数组里存放的是常量池中 abc 的地址。String str4=new String("abc") 与 str3 一样,所以 System.out.println(str3==str4); 的结果为false,而 System.out.println(str3.equals(str4)); 的运行结果为 true

(3)String str5=new String(cArr);

1
2
char [ ] cArr={'a','b','c'};
String str5=new String(cArr);

注意(2)与(3)的区别!!! (3) 里面传进去的是一个数组,不是直接给一个字符串,根据图 2 解释如下:

图2

根据上图,在栈内存里为 main 方法开辟一块空间,里面有变量 cArr 和 str5。在堆 (heap) 里开辟了一个 char(字符) 数组,把该数组的地址值赋给cArr,即cArr 指向该数组,str5 存放的是新 new 出来的对象的地址,即 str5 指向该对象,然后,重点来了,把数组的地址(cArr 的值)放到新建对象的 value[ ] 数组里(上文提到过)。然后新建对象会在堆内存里新开辟一个空间,根据地址值,找到 char [ ],把 char [ ] 数组里的内容 copy 到新开辟的空间里,然后新建对象就指向新开辟的空间。

(4)StringBuffer

1
2
StringBuffer a=new StringBuffer("A");//第一句
a.append("B");//第二句

内存图如下所示:

图3

语法第一句和文章 (2) String str1=new String("abc") 一样,在这里就不废话了。在这里补充一句,String 在底层存的是一个char [ ] 类型的数组。所以,”A” 实际存放在常量池中的 char [ ] 数组里。然后,第二句会把 “B” 放进 “A” 所在的数组里,就变成了 “AB”。假设 “A” 所在的数组存不下所要追加的内容 (在此处不成立) ,那么,会在常量池中新开一块空间,把 “A” 和所要追加的内容放里面。如图中红线所示。


我是菜鸟小七,希望我们都能成为更好的自己。